home *** CD-ROM | disk | FTP | other *** search
/ MacWorld 1999 March - Disc 1 / Macworld (1999-03) (Disk 1).dmg / Shareware World / Utilities / Text Processing / Alpha / Tcl / Modes / Perl Mode / perlEngine.tcl < prev    next >
Encoding:
Text File  |  1998-04-14  |  13.4 KB  |  557 lines  |  [TEXT/ALFA]

  1. # ◊◊◊◊ dummy to load file ◊◊◊◊ #
  2. proc perlEngine.tcl {} {}
  3.  
  4. #############################################################################
  5. # ◊◊◊◊ running scripts ◊◊◊◊ #
  6. # Tell MacPerl to run a script file:
  7. #
  8. proc perlExecuteFile {path {args {}} {flags {}}} {
  9.     global PerlmodeVars ALPHA scriptFile scriptStart filterHeadLen perlName
  10.     
  11.     if {[string length $path]} {
  12.         set perlName [file tail [app::launchBack McPL]]
  13.         if {[string length $perlName]} {
  14.                 
  15.             set ok [regexp {(.*):([^:]*)} $path pathname dirname filename]
  16.             if {!$ok} {    set name $wname    }
  17.  
  18.             if {$path != [scriptPath]} {    
  19.                 set filterHeadLen 0    
  20.             }
  21.             
  22.             if {$PerlmodeVars(perluseDebugger)} {
  23.                 append flags "debug"
  24.             }
  25.             if {$PerlmodeVars(perlpromptForArgs)} { 
  26.                 append args " [getCmdlineArgs]"
  27.             }
  28.             
  29.             sendCloseWinName $perlName $perlName
  30.             sendCloseWinName $perlName "Perl Debug"
  31.             if {$PerlmodeVars(perlautoSwitch) || $PerlmodeVars(perluseDebugger)} {
  32.                 switchTo $perlName
  33.             } else {
  34.                 message "Running file \"$filename\" as Perl script"
  35.                 watchCursor
  36.             }
  37.             
  38.             perlDoScript $perlName $path $args {} $flags
  39.             
  40. # (not sure which choice is better...)
  41. #            if {!$PerlmodeVars(perlautoSwitch)} {switchTo $ALPHA}
  42.             switchTo $ALPHA
  43. #
  44.             if {![getMacPerlError]} {
  45.                 if {$PerlmodeVars(perlretrieveOutput)} {openPerlOutput}
  46.             }
  47.         } else {
  48.             alertnote "Couldn't run MacPerl"
  49.         }
  50.     } else {
  51.         alertnote "No file specified to execute"
  52.     }
  53. }
  54.  
  55. #############################################################################
  56. # Run a MacPerl script, passed explicitly as a string:
  57. #
  58. # If no "#!/bin/perl" line already exists, one is preprended to the script
  59. # by wrapSelectScript, which also sets $filterHeadLen for use by 
  60. # getMacPerlError.
  61. proc perlExecuteScript {script {args ""} {flags {}} } {
  62.     global PerlmodeVars perlName
  63.     global scriptFile scriptStart filterHeadLen  ALPHA
  64.     
  65.     if {$script != ""} {
  66.         set script [wrapSelectScript $script]
  67.         
  68.         if {![regexp {(.*):([^:]*)} $scriptFile pathname dirname filename]} {
  69.             set filename $scriptFile 
  70.         }
  71.  
  72.         set perlName [file tail [app::launchBack McPL]]
  73.         if {[string length $perlName]} {
  74.         
  75.             if {$PerlmodeVars(perluseDebugger)} {
  76.                 append flags "debug"
  77.             }
  78.             if {$PerlmodeVars(perlpromptForArgs)} { 
  79.                 append args " [getCmdlineArgs]"
  80.             }
  81.             
  82.             sendCloseWinName $perlName $perlName
  83.             sendCloseWinName $perlName "Perl Debug"
  84.             if {$PerlmodeVars(perlautoSwitch) || $PerlmodeVars(perluseDebugger)} {
  85.                 switchTo $perlName
  86.             } else {
  87.                 message "Running buffer \"$filename\" as Perl script"
  88.                 watchCursor
  89.             }
  90.             
  91.             perlDoScript $perlName $script $args {} $flags
  92.             
  93.             switchTo $ALPHA
  94.  
  95.             if {![getMacPerlError]} {
  96.                 if {$PerlmodeVars(perlretrieveOutput)} {openPerlOutput}
  97.             }
  98.         }
  99.         
  100.     } else {
  101.             alertnote "Can't run an empty script"
  102.     }
  103. }
  104.  
  105. #############################################################################
  106. # Run a MacPerl script from the Tcl shell
  107. #
  108. # This proc pretends it is the invocation of the perl app when used 
  109. # as the first word of a command in the Tcl shell. -trf
  110. proc perl {{path {}} {args {}} } {
  111.     global PerlmodeVars ALPHA scriptFile scriptStart filterHeadLen perlName
  112.     set flags {}
  113.     
  114.     if {[string length $path]} {
  115.         set perlName [file tail [app::launchBack McPL]]
  116.         if {[string length $perlName]} {
  117.                 
  118.             set filename [file tail $path]
  119.             if {$path != [scriptPath]} {    
  120.                 set filterHeadLen 0    
  121.             }
  122.             
  123.             sendCloseWinName $perlName $perlName
  124.             sendCloseWinName $perlName "Perl Debug"
  125.             if {$PerlmodeVars(perlautoSwitch) || $PerlmodeVars(perluseDebugger)} {
  126.                 switchTo $perlName
  127.             } else {
  128.                 message "Running file \"$filename\" as Perl script"
  129.                 watchCursor
  130.             }
  131.             
  132.             perlDoScript $perlName $path $args {} $flags
  133.             
  134.             switchTo $ALPHA
  135.  
  136.             if {![getMacPerlError]} {
  137.                 if {$PerlmodeVars(perlretrieveOutput)} {openPerlOutput}
  138.             }
  139.         } else {
  140.             alertnote "Couldn't run MacPerl"
  141.         }
  142.     } else {
  143.         echo {Usage:  perl <filename> [ <args> ]}
  144.     }
  145. }
  146.  
  147.  
  148. # ◊◊◊◊ check MacPerl error msg ◊◊◊◊ #
  149. #############################################################################
  150. # Check the MacPerl output window for error messages.
  151. #
  152. proc getMacPerlError {} {
  153.     
  154.     set diag [getPerlDiag 40]
  155.     set errf [parseDiagErrf $diag]
  156.     set srcs [parseDiagSrcs $diag]
  157.     set mesg [parseDiagMesg $diag]
  158.  
  159.     if {[string length $errf]} {
  160.         showPerlDiag $diag [string length $diag] $mesg $errf $srcs
  161.         gotoPerlError $errf $srcs $mesg
  162.         return 1
  163.         
  164.     } else {
  165.         return 0
  166.     }
  167. }
  168.  
  169. #############################################################################
  170. # Check the MacPerl batch reply for error messages.
  171. #
  172. proc getBatchError {reply} {
  173.     global PerlmodeVars
  174.     set perlErrorWindow {* Perl Error Messages *}
  175.     
  176.     set fatalError 0
  177.     set diag [parseReplyDiag $reply]
  178.     set errf [parseDiagErrf  $diag ]
  179.     set srcs [parseReplySrcs $reply]
  180.     set mesg [parseDiagMesg  $diag ]
  181.     set errn [parseReplyErrn $reply]
  182.  
  183.     if {$errn} {        
  184.         showPerlDiag $diag $errn $mesg $errf $srcs
  185.         gotoPerlError $errf $srcs $mesg
  186.         set fatalError 1
  187.         
  188.     } elseif {[string length $diag] > 0} {
  189.         showPerlDiag $diag $errn $mesg $errf $srcs
  190.     }
  191.     
  192.     return $fatalError
  193. }
  194.  
  195. # ◊◊◊◊ get or show diag/errors ◊◊◊◊ #
  196.  
  197. #############################################################################
  198. # Display the Perl diagnostic output in its own window.
  199. #
  200. proc showPerlDiag {diag {errn 1} {mesg {}} {errf {}} {srcs {}}} {
  201.         global PerlmodeVars    
  202.         set perlErrorWindow {* Perl Error Messages *}
  203.         
  204.         set currWin [lindex [winNames] 0]
  205.         if {[lsearch [winNames] $perlErrorWindow] >= 0} {
  206.             bringToFront $perlErrorWindow
  207.             setWinInfo read-only 0
  208.             deleteText 0 [maxPos] 
  209.             insertText $diag
  210.         } else {
  211.             new -n $perlErrorWindow 
  212.              insertText $diag
  213.         }
  214.         
  215.         catch {shrinkWindow 2}
  216.         winReadOnly
  217.         bringToFront $currWin
  218. }
  219.  
  220. #############################################################################
  221. # Bring up a window containing the bug-ridden Perl code and highlight the
  222. # line at which the error was found.
  223. #
  224. proc gotoPerlError {errf srcs {mesg {}}} {
  225.     global PerlmodeVars scriptFile scriptStart filterHeadLen
  226.  
  227.     if {$errf == [scriptPath] || $errf == "<AppleEvent>"} {
  228.         set errf $scriptFile
  229.         # Convert it to the line number in the original file
  230.         set srcs [expr $srcs + $scriptStart - $filterHeadLen - 1]
  231.     }
  232.     # ... and leave an informative error message
  233.     #
  234.     if {[string length $mesg]} {
  235.         set mesg "$mesg at Line $srcs"            
  236.     } else {
  237.         set mesg "MacPerl flagged an error at Line $srcs"    
  238.     }
  239.     
  240.     # Bring up the script file and highlight the flagged line
  241.     #
  242.     catch {gotoFileLine $errf $srcs $mesg} fname    
  243. }
  244.  
  245. #############################################################################
  246. # Read the first block of lines (up to a maximum number) from the MacPerl
  247. # output window.
  248. #
  249. proc getPerlDiag {maxlines} {
  250.     global PerlmodeVars perlName
  251.     set pat0 {^[ \t]*$}
  252.  
  253.     set lines {}    
  254.  
  255.     # read first $maxlines of output to the MacPerl window
  256.     # (faster, but assumes error message won't appear at 
  257.     # the end of a lot of output).
  258.     #
  259.     set nlines [sendCountLines $perlName MacPerl]
  260.     set nlines [expr ($nlines > $maxlines)?$maxlines:$nlines]
  261.     if {$nlines > 0} {
  262.         set output [sendGetText $perlName $perlName 1 $nlines]
  263.         
  264.         foreach line [split $output "\r"] {
  265.             if  {[regexp $pat0 $line mtch]} {
  266.                 break
  267.             } else {
  268.                 append lines "$line\n"
  269.             }
  270.         }
  271.     }
  272.     return $lines
  273. }
  274.  
  275. # ◊◊◊◊ DoScript helpers ◊◊◊◊ #
  276.  
  277. #############################################################################
  278. # translate special DoScript flags into flags string $usrf
  279. #
  280. proc perlScriptFlags {{flags {}}} {
  281.      set usrf {}
  282.  
  283.     if {[lsearch -exact $flags "extract"] >= 0} {
  284.         append usrf { "EXTR" 'true'}
  285.     } elseif {[lsearch -exact $flags "noextract"] >= 0} {
  286.         append usrf { "EXTR" 'fals'}
  287.     }        
  288.     if {[lsearch -exact $flags "debug"] >= 0} {
  289.         append usrf { "DEBG" 'true'}
  290.     } elseif {[lsearch -exact $flags "nodebug"] >= 0} {
  291.         append usrf { "DEBG" 'fals'}
  292.     }        
  293.  
  294.     if {[lsearch -exact $flags "local"] >= 0} {
  295.         append usrf { "MODE" 'LOCL'}
  296.     } elseif {[lsearch -exact $flags "batch"] >= 0} {
  297.         append usrf { "MODE" 'BATC'}
  298.     } elseif {[lsearch -exact $flags "remote"] >= 0} {
  299.         append usrf { "MODE" 'RCTL'}
  300.     }        
  301.     return $usrf
  302.  
  303. proc perlScriptArgs {{args {}} {fileargs {}}} {
  304.     set nargs 0
  305.     set argv {}
  306.     
  307.     foreach item [parseWords $args] {
  308.         set item [string trim $item]
  309.         if {[string length $item]} {
  310.             append argv ", [curlyq $item]"
  311.             incr nargs
  312.         }
  313.     }
  314.     foreach filename $fileargs {
  315.         set item [string trim $filename]
  316.         if {[string length $item]} {
  317.             append argv ", [curlyq $item]"
  318.             incr nargs
  319.         }
  320.     }
  321.     return $argv
  322. }
  323.  
  324. #############################################################################
  325. # General Apple Event routines
  326. # (most of these have been moved to Modes:appleEvents.tcl)
  327. #
  328.  
  329.  
  330. #############################################################################
  331. # DoScript for MacPerl 4.1.3
  332. # (runs in "Local" mode under v4.1.4+)
  333. #
  334. proc perlDoScript {appname script {args {}} {fileargs {}} {flags {}} } {
  335.     # form list of quoted "command-line" args
  336.     #
  337.     if {$script != ""} {
  338.         set argv "\[[curlyq [string trim $script]]"
  339.         append argv [perlScriptArgs $args $fileargs]
  340.         append argv "]"
  341.         
  342.         set usrf [perlScriptFlags $flags]
  343.         set reply [eval "AEBuild -t 36000 -r \"$appname\" misc dosc $usrf \"----\" [list $argv] "]
  344.     #    alertnote $reply
  345.     }
  346. }
  347.  
  348. # DoScript for MacPerl 4.1.4+
  349. # [Q] do I need this for perl via shell? -trf
  350. #
  351. proc perlDoScriptBatch {appname script {args {}} {fileargs {}}} {
  352.     
  353.     # form list of quoted "command-line" args
  354.     #
  355.     if {$script != ""} {
  356.         set argv "\[[curlyq [string trim $script]]"
  357.         append argv [perlScriptArgs $args $fileargs ] 
  358.         append argv "]"
  359.                 
  360.         set reply [eval "AEBuild -t 36000 -r \"$appname\" misc dosc MODE BATC \"----\" [list $argv]"]
  361.         
  362. #         perlDisplayReply $reply
  363.  
  364.     } else {
  365.         set reply {}
  366.     }
  367.     return $reply
  368. }
  369.  
  370. # For debugging 
  371. #
  372. proc perlDisplayReply {reply} {
  373.     set currWin [lindex [winNames] 0]
  374.     new -n {*** DoScript Reply **} 
  375.     insertText $reply
  376.         
  377.     winReadOnly
  378.     catch {shrinkWindow 2}
  379.     bringToFront $currWin
  380. }
  381.  
  382. # DoScript to launch interactive debugger (for MacPerl 4.1.4+)
  383. #
  384. proc perlDoScriptDebug {appname script {args {}} {fileargs {}}} {
  385.     
  386.     # form list of quoted "command-line" args
  387.     #
  388.     if {$script != ""} {
  389.         set argv "\[[curlyq [string trim $script]]"
  390.         append argv [perlScriptArgs "$args debug" $fileargs ] 
  391.         append argv "]"
  392.                 
  393.         set reply [eval "AEBuild -t 36000 -r \"$appname\" misc dosc MODE RCTL \"----\" [list $argv]"]
  394.  
  395.         new -n {** DoScriptDebug Reply **} 
  396.         insertText $reply
  397.             
  398.         winReadOnly
  399.         catch {shrinkWindow 2}
  400.  
  401.     } else {
  402.         set reply {}
  403.     }
  404.     return $reply
  405. }
  406.  
  407. # ◊◊◊◊ parse MacPerl output ◊◊◊◊ #
  408.  
  409. #############################################################################
  410. # Extract various items out of the MacPerl diagnostic output
  411. #
  412.  
  413. # Name of the file in which the error was found
  414. #
  415. proc parseDiagErrf {diag}    {
  416.     if {![regexp {File '([^']+)'; Line} $diag allofit errf]} { 
  417.         set errf {}
  418.     }
  419.     return $errf
  420. }
  421.  
  422. # The line number on which the error was found
  423. #
  424. proc parseDiagSrcs {diag}    {
  425.     if {![regexp {File '[^']+'; Line ([0-9]+)} $diag allofit srcs]} { 
  426.         set srcs 0 
  427.     }
  428.     return $srcs
  429. }
  430.  
  431. # The error message associated with error
  432. #
  433. proc parseDiagMesg {diag} {
  434.     set pat1 {^#(.*)$}
  435.     set pat2 {File '([^']+)'; Line ([0-9]+)}
  436.     
  437.     set errMessage {}
  438.     set errFound 0
  439.     
  440.     foreach line [split $diag "\n"] {
  441.         if {[regexp $pat2 $line mtch num]} {
  442.             set errFound 1
  443.         } elseif {[regexp $pat1 $line mtch err]} {
  444.             if {$errFound == 0} {
  445.                 set errMessage $err
  446.             }
  447.         }
  448.     }
  449.     return $errMessage
  450. }
  451.  
  452. #############################################################################
  453. # Extract various return parameters out of a MacPerl DoScript reply
  454. #
  455.  
  456. # Result from batch script
  457. #
  458. proc parseReplyResult {reply} {
  459.     if {![regexp {'?\-\-\-\-'?:“([^”]*)”} $reply allofit result]} { 
  460.         set result {}
  461.     }
  462.     return $result
  463. }
  464.  
  465. # Standard output of batch script
  466. #
  467. proc parseReplyOutp {reply} {
  468.     if {![regexp {OUTP:“([^”]*)”} $reply allofit outp]} { 
  469.         set outp {}
  470.     }
  471.     return $outp
  472. }
  473.  
  474. # Diagnostic output of the batch script
  475. #
  476. proc parseReplyDiag {reply}    {
  477.     if {[regexp {diag:“([^”]*)”} $reply allofit diag]}  {
  478.     } else { 
  479.         set diag {}
  480.     }
  481.     return $diag
  482. }
  483.  
  484. # File alias of the script file in which the error was found
  485. #
  486. proc parseReplyErob {reply}    {
  487.     if {![regexp {erob:alis\(«(.*)»\)} $reply allofit erob]} {
  488.         set erob {} 
  489.     }
  490.     return $erob
  491. }
  492.  
  493. # First line flagged in error
  494. #
  495. proc parseReplySrcs {reply}    {
  496.     if {![regexp {erng:{srcs:([0-9]+)[^\}]*}} $reply allofit srcs]} { 
  497.         set srcs 0 
  498.     }
  499.     return $srcs
  500. }
  501.  
  502. # Last line flagged in error
  503. #
  504. proc parseReplySrce {reply}    {
  505.     if {![regexp {erng:{[^\}]*srce:([0-9]+)}} $reply allofit srce]} { 
  506.         set srce 0
  507.     }
  508.     return $srce
  509. }
  510.  
  511. # Error number
  512. #
  513. proc parseReplyErrn {reply}    {
  514.     if {![regexp {errn:([0-9]+)} $reply allofit errn]} {
  515.         set errn 0
  516.     }
  517.     return $errn
  518. }
  519.  
  520. #############################################################################
  521. # Read the MacPerl output window and load the contents, if any, into
  522. # a new Alpha window. 
  523. # Modified to direct output to Tcl Shell if perl was called from there -trf
  524. #
  525. proc openPerlOutput {} {
  526.     global PerlmodeVars perlRecycleOutput perlName
  527.     set perlOutputWindow {* Perl Output *}
  528.     
  529.     set output [sendGetText $perlName $perlName]
  530.     if {[string length $output]} {
  531.         if {[win::CurrentTail] == "*tcl shell*"} {
  532.             endOfBuffer
  533.             insertText \r $output
  534.             endOfBuffer 
  535.         } elseif {$PerlmodeVars(perlRecycleOutput) && 
  536.             [lsearch [winNames] $perlOutputWindow] >= 0} {
  537.             
  538.             bringToFront $perlOutputWindow
  539.             replaceText [minPos] [maxPos] $output
  540.             catch {shrinkWindow 2}
  541.             setWinInfo dirty 0
  542.             goto [minPos]
  543.         } else {
  544.             new -n $perlOutputWindow
  545.             insertText $output
  546.             catch {shrinkWindow 2}
  547.             setWinInfo dirty 0
  548.             goto [minPos]
  549.         }
  550.     }
  551. }
  552.